//
//  XTBaseViewController.swift
//  MyBlog
//
//  Created by victor on 16/11/10.
//  Copyright © 2016年 victor. All rights reserved.
//

import UIKit

// 面试题：oc中支持多继承吗？如果不支持如何替代？使用协议替代。(协议里是方法，遵守了协议，实现了方法就具备了这个能力)
// swift 写法更类似于多继承
// class XTBaseViewController: UIViewController,UITableViewDataSource,UITableViewDelegate {
// swift中，利用extension 可以把`函数`按照功能分类管理，便于阅读和维护
// 注意
// 1. extension 中不能有属性
// 2. extension 中不能重写`父类`本类的(扩展中的方法可以重写)方法，重写父类方法，是子类的职责，扩展是对类的扩展
// 注意：父类方法写在extension里，子类方法也要写在extension里，否则可能会出现代码不执行的情况

/// 所有主控制器的基类控制器
class XTBaseViewController: UIViewController {
    
    /// 访客视图信息字典
    var vistorInfoDictionary : [String: String]?
    
    /// 表格视图 - 如果用户没有登录就不创建，因此这里不用用懒加载
    var tableView : UITableView?
    /// 刷新控件
    var refreshControl : XTRefreshControl = XTRefreshControl()
    /// 上拉刷新标记
    var isPullup = false
    
    /// 自定义导航条
    lazy var navigationBar = UINavigationBar(frame:CGRect(x: 0, y: 0, width: UIScreen.cz_screenWidth(), height: 64))
    
    /// 自定义导航条目 - 以后设置导航栏内容，统一使用 navItem
    lazy var navItem = UINavigationItem()

    override func viewDidLoad() {
        super.viewDidLoad()
        setupUI()
        
        XTNetWorkManger.shared.userLogon ? loadData() : ()
        
        // 注册通知
        NotificationCenter.default.addObserver(self, selector: #selector(loginSuccess), name: NSNotification.Name(rawValue: XTUserLoginSuccessedNotification), object: nil)
    }
    
    deinit {
        // 注销通知
        NotificationCenter.default.removeObserver(self)
    }
    
    /// 重写title的 setter方法
    override var title: String? {
        didSet {
            navItem.title = title
        }
    }
    
    /// 加载数据 - 具体的实现由子类负责
    func loadData() {
        // 如果子类不实现任何方法，默认关闭刷新控件
        refreshControl.endRefreshing()
    }
}

//MARK: -访客视图监听方法
extension XTBaseViewController {
    
    /// 登录成功处理
    @objc fileprivate func loginSuccess(n: Notification) {
        print("登录成功 \(n)")
        
        // 登录前左边是注册，右边是登录
        navItem.leftBarButtonItem = nil
        navItem.rightBarButtonItem = nil
        
        // 更新UI -> 将访客视图替换为表格视图
        // 需要重新设置 view
        // 在访问 view 的 getter 时，如果 view == nil 会调用 loadView -> viewDidLoad
        // 这样就相当于重新调用了一遍setupUI() 方法
        view = nil
        
        // 注销通知 -> 重新执行 viewDidLoad 会再次注册！避免通知被重复注册
        NotificationCenter.default.removeObserver(self)
    }

    @objc fileprivate func login() {
        // 发送通知
        NotificationCenter.default.post(name: NSNotification.Name(rawValue: XTUserShouldLoginNotification), object: nil)
    }
    
    @objc fileprivate func register() {
        print("用户注册")
    }
}

//MARK: -设置界面
extension XTBaseViewController {
    fileprivate func setupUI() {
        view.backgroundColor = UIColor.white
        
        // 当屏幕上有navigationBar 和 ScrollView的时候要注意考虑缩进的问题
        // 取消自动缩进 - 如果隐藏了导航栏，会缩进 20 个点
        automaticallyAdjustsScrollViewInsets = false
        
        setupNavigationBar()
        XTNetWorkManger.shared.userLogon ? setupTableView() : setupVistorView()
    }
    
    /// 设置表格视图 - 用户登录之后执行
    /// 子类重写此方法，因为子类不需要关心用户登录之前的逻辑
    func setupTableView() {
        tableView = UITableView(frame: view.bounds, style: .plain)
        // 将tableView 设置在navigationBar下面，就不用在意设置navigationBar 和 tableView 的顺序问题了
        view.insertSubview(tableView!, belowSubview: navigationBar)
        
        // 设置数据源和代理 -> 目的，子类直接实现数据源方法
        tableView?.dataSource = self
        tableView?.delegate = self
        
        // 设置内容缩进
        tableView?.contentInset = UIEdgeInsets(top: navigationBar.bounds.height,
                                               left: 0,
                                               bottom: tabBarController?.tabBar.bounds.height ?? 49,
                                               right: 0)
        
        // 修改指示器的缩进 -强行解包是为了拿到一个必有的 inset
        tableView?.scrollIndicatorInsets = tableView!.contentInset
        
        // 设置刷新控件
        refreshControl = XTRefreshControl()
        
        // 添加到表格视图
        tableView?.addSubview(refreshControl)
        
        // 添加监听方法
        refreshControl.addTarget(self, action: #selector(loadData), for: .valueChanged)
    }
    
    /// 设置访客视图
    fileprivate func setupVistorView() {
        let vistorView = XTVistorView(frame:view.bounds)
        
        // 这种情况会导致导航栏被挡住
        //view.addSubview(vistorView)
        view.insertSubview(vistorView, belowSubview: navigationBar)
        
        // 设置访客视图信息
        vistorView.vistorInfo = vistorInfoDictionary
        
        // 添加访客视图按钮坚挺方法
        vistorView.loginBtn.addTarget(self, action: #selector(login), for: .touchUpInside)
        vistorView.registerBtn.addTarget(self, action: #selector(register), for: .touchUpInside)
        
        // 设置导航条按钮
        navItem.leftBarButtonItem = UIBarButtonItem(title: "注册", style: .plain, target: self, action: #selector(register))
        navItem.rightBarButtonItem = UIBarButtonItem(title: "登录", style: .plain, target: self, action: #selector(login))
    }
    
    /// 设置导航条
    fileprivate func setupNavigationBar() {
        //添加导航条
        view.addSubview(navigationBar)
        //将item设置给bar
        navigationBar.items = [navItem]
        // 设置navBar 整个背景的渲染颜色 因为系统默认的导航栏透明度太高(如果不设置会看到点击的时候有个地方特别亮的感觉)
        navigationBar.barTintColor = UIColor.cz_color(withHex: 0xF6F6F6)
        // 设置navBar的字体颜色
        navigationBar.titleTextAttributes = [NSForegroundColorAttributeName:UIColor.darkGray]
        // 设置系统按钮的文字渲染颜色
        navigationBar.tintColor = UIColor.orange
    }
}

//MARK: -UITableViewDataSource,UITableViewDelegate
extension XTBaseViewController: UITableViewDataSource,UITableViewDelegate {
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 0
    }
    
    // 基类只是准备方法，子类负责具体实现
    // 子类的数据源方法不需要super
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        // 下面这句只是保证没有语法错误
        return UITableViewCell()
    }
    
    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        return 10
    }
    
    // 在显示最后一行的时候，做上拉刷新
    func tableView(_ tableView: UITableView, willDisplay cell: UITableViewCell, forRowAt indexPath: IndexPath) {
        //判断 indexPath 是否是最后一行 (indexPath.section(最大) / indexPath.row(最后一行))
        let row = indexPath.row
        
        let section = tableView.numberOfSections - 1
        
        // 如果没有数据 就直接返回
        if row < 0 || section < 0 {
            return
        }
        
        // 行数
        let count = tableView.numberOfRows(inSection: section)
        
        // 如果是最后一行，同时没有开始上拉刷新
        if row == (count - 1) && !isPullup {
            print("上拉刷新")
            
            isPullup = true
            
            // 开始刷新
            loadData()
        }
        
    }
}
